blk_stop_queue(info->rq);
/* No more gnttab callback work. */
gnttab_cancel_free_callback(&info->callback);
- flush_scheduled_work();
spin_unlock_irqrestore(&blkif_io_lock, flags);
+ /* Flush gnttab callback work. Must be done with no locks held. */
+ flush_scheduled_work();
+
xlvbd_del(info);
xenbus_frontend_closed(dev);
blk_stop_queue(info->rq);
/* No more gnttab callback work. */
gnttab_cancel_free_callback(&info->callback);
- flush_scheduled_work();
spin_unlock_irq(&blkif_io_lock);
+ /* Flush gnttab callback work. Must be done with no locks held. */
+ flush_scheduled_work();
+
/* Free resources associated with old device channel. */
if (info->ring_ref != GRANT_INVALID_REF) {
gnttab_end_foreign_access(info->ring_ref, 0,